

# **Avnet Technical Training Course**



Tools: 2019.1

Training Version: v11

Date: July 2019

© 2019 Avnet. All rights reserved. All trademarks and registered trademarks are the property of their respective owners. All specifications are subject to change without notice

NOTICE OF DISCLAIMER: Avnet is providing this design, code, or information "as is." By providing the design, code, or information as one possible implementation of this feature, application, or standard, Avnet makes no representation that this implementation is free from any claims of infringement. You are responsible for obtaining any rights you may require for your implementation. Avnet expressly disclaims any warranty whatsoever with respect to the adequacy of the implementation, including but not limited to any warranties or representations that this implementation is free from claims of infringement and any implied warranties of merchantability or fitness for a particular purpose.



# Lab 4 Overview

With a Hardware Platform and BSP, we are now ready to add an application. With the BSP generated and built, drivers for all the peripherals are now ready and available in the workspace. In fact, Xilinx provides example code for most of the peripherals! We are ready to start writing our own code!

In addition to creating a brand new application, we will also take advantage of two auto-generated, example applications that will allow us to see much more sophisticated software applications than we would have time to develop during this course.

# **Lab 4 Objectives**

When you have completed Lab 4, you will know how to:

- Add new software applications to SDK
- Use example code to target the UART in a Hello World application
- Apply example project templates, including
  - Memory Tests
  - o Peripheral Tests
- Identify application code size and location
- Modify linker scripts to change the target memory location





# Experiment 1: Develop an Application with Example Code

SDK is now ready for software development. Getting started can be difficult. You may have inherited a hardware platform that you are still working to understand. You may not be familiar with the drivers that Xilinx has provided in the generated BSP. Reading through all the driver code to get started could be overwhelming. It would be extremely helpful to start with example code and then make modifications.

That is exactly what we will do in this experiment. Hopefully this will help you get in the habit of looking for free example code provided by Xilinx any time you encounter a new peripheral and driver.

#### **Experiment 1 General Instruction:**

Add an Empty software application. Copy and paste the Hello World example code provided with the UART driver. Generate the linker script to target the application memory region to the PS7 on-chip RAM0. Determine the code size and location.

#### **Experiment 1 Step-by-Step Instructions:**

- 1. In SDK, select File → New → Application Project.
- 2. In the **Project Name** field type in Hello\_Zynq. Change the **BSP** to the existing Standalone BSP. Click **Next** >.



Figure 1 - New Application Wizard





3. You can see in the *Available Templates* that there is a Hello World template. We are NOT going to use this right now, although it would work perfectly fine. Instead, select **Empty Application**. Click **Finish**.



Figure 2 - Empty Application

4. Notice that the Hello\_Zynq application is now visible in *Project Explorer*. Notice the different icons that represent C Software Application , BSP , and Hardware Platform



Figure 3 - Project Explorer View with Hello World C Application Added





5. In *Project Explorer*, browse to **Hello\_Zynq**  $\rightarrow$  **src**. You will see that you have been given two files of note. The file lscript.ld is a default linker script. The README.txt states

Empty application. Add your own sources.

 As instructed, we will add our own source now. Right-click on the src folder, then select New → Source File. In the dialog, give the source file the name hello\_zynq.c, then click Finish.



Figure 4 - New Source File

7. By default, SDK will build the application automatically after it is added. This can be changed by deselecting **Project > Build Automatically**, but we will not do that now. The results of the build are available in the Console window at the bottom of the SDK GUI. In this case, the build fails because the new hello\_zynq.c has nothing in it other than a commented header.

```
CDT Build Console [Hello_Zynq]
a9-linaro-pre-build-step

Building file: ../src/hello_zynq.c
Invoking: ARM v7 gcc compiler
arm-none-eabi-gcc -Wall -00 -g3 -c -fmessage-length=0 -MT"src/hello_zynq.o" -mcpu=cortex-a9 -mfpu=vfpv3 -mfloat-abi=hard -I../../standalone_bsp_0/ps7_Finished building: ../src/hello_zynq.c

Building target: Hello_Zynq.elf
Invoking: ARM v7 gcc linker
arm-none-eabi-gcc -mcpu=cortex-a9 -mfpu=vfpv3 -mfloat-abi=hard -Wl,-build-id=none -specs=Xilinx.spec -Wl,-T -Wl,../src/lscript.ld -L../../standalone_br/home/training/SDK/2019.1/gnu/aarch32/lin/gcc-arm-none-eabi/bin/../lib/gcc/arm-none-eabi/8.2.0/../../arm-none-eabi/bin/ld: ../../standalone_bsp_0/makefile:39: recipe for target 'Hello_Zynq.elf' failed
/home/training/Annetraining/Annetraining/SDK/201911/SDK Workspace/standalone_bsp_0/ps7_cortexa9_0/libsrc/standalone_v7_0/src/xil-crt0.5:138: undefined reference to collect2: error: ld returned 1 exit status
make: *** [Hello_Zynq.elf] Error 1

10:20:07 Build Finished (took 421ms)
```

Figure 5 - Auto Build of Hello\_Zynq Fails

8. Another artifact of the automatic build is that the BSP now has an error because it couldn't find a main() in the new application. You will see the BSP icon with a red X on it now. We'll fix this in a minute after we have a main().



Figure 6 – BSP Error Since the New Source has no main()





- 9. Next, we'll locate some example code. Using a file explorer navigate to
  - /home/training/SDK/2019.1/data/embeddedsw/XilinxProcessorIPLib/drivers/uartps\_v3\_7/examples.
- 10. This opens the directory where the example source files are located. xuartps\_hello\_world\_example.c is provided, along with several other examples that would likely prove useful in future development including polled, interrupt, and echo examples. Right Click on xuartps\_hello\_world\_example.c. and open the file using Open with → gedit



Figure 7 - Example UART C Files Driver Directory

- 11. Use Ctrl-A or click-drag to select the contents of the entire file. Then use Ctrl-C to copy. Switch back to SDK, close the import examples window and use Ctrl-V to paste into hello\_zynq.c. Take a moment to browse through the code to understand what it is doing.
- 12. Use **File** → **Save**, or the Save icon , or Ctrl-S to save the file. The application automatically builds.



Figure 8 - Hello World Application Automatically Built

13. Notice at the bottom of the report that the application size is reported. This is printed from the report file Hello\_Zynq.elf.size which can also be found in *Project Explorer* under Hello\_Zynq → Debug → Hello\_Zynq.elf.size. Note that "Debug" refers to the Build Configuration, which by default is always Debug for a new application. Note that this application is 56,232 bytes large.

Be aware that you may see small deviations to these sizes reported as it is dependent on the exact compiler version and options you are using.







Figure 9 – Size of the Hello\_Zynq Application

14. Now that we have a main() in the new application, we can correct the issue with the BSP. SDK should be able to fix itself, but in case it doesn't, do the following. Select standalone\_bsp\_0 in *Project Explorer*. Now select **Project → Clean**. Select the radio button for *Clean projects selected below*. Check the box for standalone\_bsp\_0. Click OK. After cleaning, an automatic rebuild will start which will eliminate the red X from the BSP.



Figure 10 - Clean and Rebuild the BSP

- 15. The Iscript.Id is the linker script for this application. This is useful as a report to see what memory was targeted for the application. It may also be edited to change the location of the application. Double-click **Hello\_Zynq → src → Iscript.Id** now. Notice that the window opens to a graphical view that is indicated by the *Summary* tab in the lower left corner. We'll look at the *Source* tab in a minute.
- 16. The first Summary section shows the Available Memory Regions.
  - a. axi\_bram\_ctrl\_0\_Mem0 is blockRAM located in the PL
  - b. **ps7\_ddr\_0** is the DDR3L
  - c. ps7\_qspi\_linear\_0 is the QSPI
  - d. ps7\_ram\_0 is 192 KB of on-chip RAM
  - e. ps7\_ram\_1 is 64 KB of on-chip RAM





#### **Available Memory Regions**

| Name                 | Base Address | Size       |
|----------------------|--------------|------------|
| axi_bram_ctrl_0_Mem0 | 0x40000000   | 0x2000     |
| ps7_ddr_0            | 0x100000     | 0x1FF00000 |
| ps7_qspi_linear_0    | 0xFC000000   | 0x1000000  |
| ps7_ram_0            | 0x0          | 0x30000    |
| ps7_ram_1            | 0xFFFF0000   | 0xFE00     |

Figure 11 – Available Memory Regions for Application

17. The second *Summary* section shows the *Stack and Heap Sizes*. The default size is 8KB each for Stack and Heap.



Figure 12 - Stack and Heap Sizes

18. The third *Summary* section shows the mapping between the application Sections and the Memory Region. You'll see in this case that all 29 different sections were all assigned to DDR3L.





#### Section to Memory Region Mapping

| Section Name      | Memory Region |
|-------------------|---------------|
| .text             | ps7_ddr_0     |
| .init             | ps7_ddr_0     |
| .fini             | ps7_ddr_0     |
| .rodata           | ps7_ddr_0     |
| .rodata1          | ps7_ddr_0     |
| .sdata2           | ps7_ddr_0     |
| .sbss2            | ps7_ddr_0     |
| .data             | ps7_ddr_0     |
| .data1            | ps7_ddr_0     |
| .got              | ps7_ddr_0     |
| .ctors            | ps7_ddr_0     |
| .dtors            | ps7_ddr_0     |
| .fixup            | ps7_ddr_0     |
| .eh_frame         | ps7_ddr_0     |
| .eh_framehdr      | ps7_ddr_0     |
| .gcc_except_table | ps7_ddr_0     |
| .mmu_tbl          | ps7_ddr_0     |
| .ARM.exidx        | ps7_ddr_0     |
| .preinit_array    | ps7_ddr_0     |
| .init_array       | ps7_ddr_0     |
| .fini_array       | ps7_ddr_0     |
| .ARM.attributes   | ps7_ddr_0     |
| .sdata            | ps7_ddr_0     |
| .sbss             | ps7_ddr_0     |
| .tdata            | ps7_ddr_0     |
| .tbss             | ps7_ddr_0     |
| .bss              | ps7_ddr_0     |
| .heap             | ps7_ddr_0     |
| .stack            | ps7_ddr_0     |

Figure 13 – Section to Memory Region Mapping

This *Summary* is editable. You could easily change the Stack and Heap sizes here. You could also change the *Memory Region* for one section at a time. If you wanted to change the target Memory Region for all sections, this would be very tedious here.

19. Switch to the Source tab.





Figure 14 - Source for Linker Script

Now you see the source linker script *code* from which the Summary was generated. This view is also editable. If you wanted to change all sections to the on-chip RAM, you could do a global search and replace here. That is straight-forward, and many will be comfortable editing the linker script in this manner.

20. Close Iscript.ld.





21. Another method to modify the linker script is to generate a completely new one using a wizard. This is possible with a tool provided in the SDK. Right-click on **Hello\_Zynq** in the *Project Explorer* and select **Generate Linker Script**.



Figure 15 – Generate Linker Script





22. Notice that in the *Hardware Memory Map*, you see the same five memories that were in the Iscript.ld *Summary*.



Figure 16 - Hardware Memory Map

23. On the right side in the *Basic* tab, set all of the sections to *ps7\_ram\_0*. This Linker Script Generator defaults to 1 KB for *Heap Size* and *Stack Size* which is smaller than the 8KB created when Hello\_Zynq was generated, but 1KB is acceptable. If you wanted to change this, type in the number of bytes (for example, 2048 rather than 2 KB). The *Advanced* tab allows you to be much more specific about assigning particular pieces within the Code and Data section, but we won't do that today. Click **Generate** and then **Yes** to allow overwriting the existing linker script.



Figure 17 – Generate Linker Script for Memory Tests

- 24. Notice that SDK will automatically rebuild the Hello\_Zynq application based on the new linker script. The new size is ~41,896bytes. It makes sense since both the stack and heap were each reduced by 7 KB each.
- 25. Open the newly generated lscript.ld to see the changes implemented by the Generate Linker Script tool.





26. Close Iscript.ld and any other files related to Hello\_Zynq.

|     |        |               |     | - 2 1 |               |    |   |
|-----|--------|---------------|-----|-------|---------------|----|---|
| / I | , ,    |               | 0   | +1    |               | n  |   |
| w   | ,      | е             | . ` | "     | O             | 14 |   |
| ~   | $\sim$ | $\overline{}$ | _   | •     | $\overline{}$ |    | • |
|     | _      |               |     |       |               |    | _ |

| Answer the following question:                                                                                                                            |      |  |
|-----------------------------------------------------------------------------------------------------------------------------------------------------------|------|--|
| You've been assigned a task to develop code to test reading and writing to the PL (peripheral axi_bram_ctrl_0 in this hardware platform). What do you do? | BRAM |  |
|                                                                                                                                                           |      |  |
|                                                                                                                                                           |      |  |
|                                                                                                                                                           |      |  |
|                                                                                                                                                           |      |  |
|                                                                                                                                                           |      |  |





# **Experiment 2: Add Peripheral Test**

It would be nice to test the various peripherals in the hardware platform. SDK provides a template for testing the peripherals. This peripheral test exercises all of the peripherals and their associated controllers built inside the SoC.

#### **Experiment 2 General Instruction:**

Add the Peripheral Test application. Determine the size and target location.

#### **Experiment 2 Step-by-Step Instructions:**

- In SDK, select File → New → Application Project.
- 2. In the Project Name field type in Test\_Peripherals. Change the BSP to the existing standalone\_bsp\_0. By default, a new application project selects the option to generate its own BSP. If we allowed the tool to do this, the new BSP will be identical to the one already created. If at any point we wanted to change a setting in the BSP, we would have to change it in multiple BSPs, and all of them would have to rebuild. It is much better to have only one BSP, unless you are purposely making unique ones for a reason. Click Next >.



Figure 18 - New Application Wizard





3. Select Peripheral Tests from the Available Templates field. Click Finish.



Figure 19 - New Application Project: Peripheral Tests





#### Questions:

| Answer the following questions: |                                                                        |  |
|---------------------------------|------------------------------------------------------------------------|--|
| •                               | To what memory region(s) is the Test_Peripherals application targeted? |  |
| _                               |                                                                        |  |

4. The Peripheral Test application has an example of enabling the I- and D-caches. Open source file testperiph.c, which is the source file containing main(). Right click in the left margin of the file and select show line numbers.



Figure 20 - Toggle Show Line Numbers

5. Notice the following code on lines 63 & 64:

```
Xil_ICacheEnable();
Xil_DCacheEnable();
```

6. You might wonder if these enable functions are enabling the L1 cache, the L2 cache, or both. Select one of them, then right-click and select **Open Declaration**.



Figure 21 – Open Cache Declaration

7. The function declaration is displayed in source code file xil\_cache.c, which is part of the BSP. You'll notice that both CacheEnable() functions include the commands to enable both the L1 and L2 caches.





```
557⊖ void Xil_ICacheEnable(void)

558 {

559  Xil_L1ICacheEnable();

560  #ifndef USE_AMP

561  Xil_L2CacheEnable();

562  #endif

563 }
```

Figure 22 – Declaration Shows Both L1 and L2 Enabled





# **Experiment 3: Add and Edit Memory Test**

Another useful application template that the SDK provides is a Memory Test. This is a very useful test for any new hardware system to make sure the memory is stable prior to running an O/S.

#### **Experiment 3 General Instruction:**

Add the Memory Test application. Determine the size and target location. Copy the project. Edit the copied Memory Test to expand the test region to the entire memory.

## **Experiment 3 Step-by-Step Instructions:**

1. Repeat steps 1 through 3 of Experiment 2 to add a new application, called **Test\_Memory** with the *Memory Test*s template applied.



Figure 23 – Adding New Application Project Test\_Memory







Figure 24 – Apply the Memory Tests Template

Use Iscript.ld files to answer the following question.

## Question:

# Answer the following question:

- To what memory region(s) is the Test\_Memory application targeted?
- 2. Open the memorytest.c source file. In main(), you will notice that a *for* loop exercises function test\_memory\_range() across n\_memory\_ranges memories.
- 3. Inside the test\_memory\_range() declaration, you'll see that three different tests are run one 32-bit test, one 16-bit test, and one 8-bit test. Hover over the Xil\_TestMem8() call to see the function prototype. Notice that the second parameter is the number of words to be tested.





4. Inside main() Hover over n\_memory\_ranges and memory\_ranges[] (around line 100) to see what values are assigned. Note: you can click into the hover pop-up window to scroll down if the contents are not completely displayed.

## Questions:

| Questions.                                                   |  |  |
|--------------------------------------------------------------|--|--|
| Answer the following questions:                              |  |  |
| How much memory is tested by default?                        |  |  |
| How many memories are tested? Which ones? Which is not? Why? |  |  |
|                                                              |  |  |





5. Copy the entire Test\_Memory application project by right-clicking and selecting **Copy**. Then right-click again in the whitespace underneath the *Project Explorer* and select **Paste**.



Figure 25 – Copy Test\_Memory



Figure 26 - Paste Test\_Memory





6. Type Test\_Memory\_FullDDR as the Project name then click OK.



Figure 27 - Copy Project

- 7. Expand the *Test\_Memory\_FullDDR* application in the Project Explorer and then the *src* folder. Double-click on memorytest.c.
- 8. Scroll to where function test\_memory\_range is defined. This function tests the same 4096-byte chunk three different ways: 1024 words in 32-bit mode, 2048 half-words in 16-bit mode, and 4096 bytes in 8-bit mode. Since the memory is 512 MB, you could modify the code to test the full region. To illustrate increasing the test window (although not to the full 512 MB region in the interest of test time), change the code to test 1 MB (1,048,576), as shown below. Save the code using Ctrl-S on your keyboard. The application will automatically rebuild.

```
status = Xil_TestMem32((u32*)range->base, 1048576/4, 0xAAAA5555,
XIL_TESTMEM_ALLMEMTESTS);

status = Xil_TestMem16((u16*)range->base, 1048576/2, 0xAA55,
XIL_TESTMEM_ALLMEMTESTS);

status = Xil_TestMem8((u8*)range->base, 1048576, 0xA5,
XIL TESTMEM_ALLMEMTESTS);
```





- 9. Since the on-chip RAMs are much less than 1 MB, we don't want to run a 1 MB test those RAMs. Edit memory config q.c as this:
- Move the ps7\_ddr\_0 section to the first range in the memory\_range\_s structure :

- set **n\_memory\_ranges** to 1 rather than 3. Save the file.

# **Exploring Further**

If you have more time and would like to investigate more...

• Examine the remainder of the Test Peripherals source code to determine which peripherals will be tested.

This concludes Lab 4.





**Revision History** 

| Date      | Version | Revision                                          |
|-----------|---------|---------------------------------------------------|
|           |         |                                                   |
| 12 Nov 13 | 01      | Initial release                                   |
| 23 Nov 13 | 02      | Revisions after pilot                             |
| 01 May 14 | 03      | ZedBoard.org Training Course Release              |
| 30 Oct 14 | 04      | Revised for Vivado 2014.3                         |
| 31 Dec 14 | 05      | Revised for Vivado 2014.4                         |
| 12 Mar 15 | 06      | Finalize for SDK 2014.4                           |
| Oct 15    | 07      | Updated to Vivado 2015.2                          |
| Aug 16    | 08      | Updated to Vivado 2016.2                          |
| Jun 17    | 09      | Updated to Vivado 2017.1 for MiniZed + Rebranding |
| Jan 18    | 10      | Updated to Vivado/SDK 2017.4                      |
| July 19   | 11      | Updated to Vivado/SDK 2019.1                      |

# Resources

www.minized.org

www.microzed.org

www.picozed.org

www.zedboard.org

www.xilinx.com/zynq

www.xilinx.com/sdk

www.xilinx.com/vivado

www.xilinx.com/support/documentation/sw manuals/ug949-vivado-design-methodology.pdf

www.xilinx.com/support/documentation/sw\_manuals/ug1046-ultrafast-design-methodologyguide.pdf





# **Answers**

## **Experiment 1**

 You've been assigned a task to develop code to test reading and writing to the PL BRAM (peripheral axi\_bram\_ctrl\_0 in this hardware platform). What do you do?

When starting to work with a new peripheral and its associated driver, the best place to start is the example code provided by Xilinx. Go to the system.mss Overview. Find the BRAM peripheral. Click on the Examples to get to the example code.

## **Experiment 2**

• To what memory region(s) is the Test\_Peripherals application targeted?

DDR3 for everything – Code, Data, heap, and stack

### **Experiment 3**

• To what memory region(s) is the Test\_Memory application targeted?

ps7\_ram\_0\_S\_AXI\_BASEADDR for the Code, Data, heap, and stack sections

• How much memory is tested by default?

4096 bytes

How many memories are tested? Which ones? Which is not, Why?

Three memories are tested: DDR3, axi\_bram, and ram\_1.

ram\_0 is not tested as this is where the code, data, heap and stack reside.

